import type { UserInfo } from '/#/store';
import { defineStore } from 'pinia';
import { store } from '/@/store';
import { RoleEnum } from '/@/enums/roleEnum';
import { PageEnum } from '/@/enums/pageEnum';
import { ROLES_KEY, TOKEN_KEY, USER_INFO_KEY } from '/@/enums/cacheEnum';
import { getAuthCache, setAuthCache } from '/@/utils/auth';
import { GetUserInfoModel, LoginParams } from '/@/api/system/model/userModel';
import { doLogout, getUserInfo, loginApi } from '/@/api/system/userManagement';
import { useI18n } from '/@/hooks/web/useI18n';
import { useMessage } from '/@/hooks/web/useMessage';
import { router } from '/@/router';
import { usePermissionStore } from '/@/store/modules/permission';
import { RouteRecordRaw } from 'vue-router';
import { PAGE_NOT_FOUND_ROUTE } from '/@/router/routes/basic';

interface UserState {
    userInfo: Nullable<UserInfo>;
    token?: string;
    roleList: RoleEnum[];
    sessionTimeout?: boolean;
}

export const useUserStore = defineStore({
    id: 'app-user',
    state: (): UserState => ({
        // user info
        userInfo: null,
        // token
        token: undefined,
        // roleList
        roleList: [],
        // Whether the login expired
        sessionTimeout: false,
    }),
    getters: {
        getUserInfo(): UserInfo {
            return this.userInfo || getAuthCache<UserInfo>(USER_INFO_KEY) || {};
        },
        getToken(): string {
            return this.token || getAuthCache<string>(TOKEN_KEY);
        },
        getRoleList(): RoleEnum[] {
            return this.roleList.length > 0 ? this.roleList : getAuthCache<RoleEnum[]>(ROLES_KEY);
        },
        getSessionTimeout(): boolean {
            return !!this.sessionTimeout;
        },
    },
    actions: {
        setToken(info: string | undefined) {
            this.token = info;
            setAuthCache(TOKEN_KEY, info);
        },
        setRoleList(roleList: RoleEnum[]) {
            this.roleList = roleList;
            setAuthCache(ROLES_KEY, roleList);
        },
        setUserInfo(info: UserInfo) {
            this.userInfo = info;
            setAuthCache(USER_INFO_KEY, info);
        },
        setSessionTimeout(flag: boolean) {
            this.sessionTimeout = flag;
        },
        resetState() {
            this.userInfo = null;
            this.token = '';
            this.roleList = [];
            this.sessionTimeout = false;
        },
        /**
         * @description: login
         */
        async login(
            params: LoginParams & {
                goHome?: boolean;
            }
        ): Promise<GetUserInfoModel | null> {
            try {
                const { goHome = true, ...loginParams } = params;
                const data = await loginApi(loginParams);
                const { token } = data;

                // save token
                this.setToken(token);
                // get user info
                const userInfo = await this.getUserInfoAction();

                const sessionTimeout = this.sessionTimeout;
                if (sessionTimeout) {
                    this.setSessionTimeout(false);
                } else if (goHome) {
                    const permissionStore = usePermissionStore();
                    if (!permissionStore.isDynamicAddedRoute) {
                        const routes = await permissionStore.buildRoutesAction();
                        routes.forEach((route) => {
                            router.addRoute(route as unknown as RouteRecordRaw);
                        });
                        router.addRoute(PAGE_NOT_FOUND_ROUTE as unknown as RouteRecordRaw);
                        permissionStore.setDynamicAddedRoute(true);
                    }
                    await router.replace(PageEnum.BASE_HOME);
                }
                return userInfo;
            } catch (error) {
                return Promise.reject(error);
            }
        },
        async getUserInfoAction(): Promise<UserInfo> {
            const userInfo = await getUserInfo();
            const { roles } = userInfo;
            const roleList = roles.map((item) => item.role_code) as RoleEnum[];
            this.setUserInfo(userInfo);
            this.setRoleList(roleList);
            return userInfo;
        },
        /**
         * @description: logout
         */
        async logout(goLogin = false) {
            try {
                await doLogout();
            } catch {
                console.log('注销Token失败');
            }
            this.setToken(undefined);
            this.setSessionTimeout(false);
            goLogin && router.push(PageEnum.BASE_LOGIN);
        },

        /**
         * @description: Confirm before logging out
         */
        confirmLoginOut() {
            const { createConfirm } = useMessage();
            const { t } = useI18n();
            createConfirm({
                iconType: 'warning',
                title: t('sys.app.logoutTip'),
                content: t('sys.app.logoutMessage'),
                onOk: async () => {
                    await this.logout(true);
                },
            });
        },
    },
});

// Need to be used outside the setup
export function useUserStoreWithOut() {
    return useUserStore(store);
}
